Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Fix absolute path check for Windows #15235

Merged
merged 1 commit into from
Aug 8, 2024
Merged

Fix absolute path check for Windows #15235

merged 1 commit into from
Aug 8, 2024

Conversation

brandonchinn178
Copy link
Contributor

Summary

Fixes #15109, regression introduced in #12519

Test plan

Copy link

netlify bot commented Aug 3, 2024

Deploy Preview for jestjs ready!

Built without sensitive environment variables

Name Link
🔨 Latest commit 3835811
🔍 Latest deploy log https://app.netlify.com/sites/jestjs/deploys/66b2e56d18981e0008ec348f
😎 Deploy Preview https://deploy-preview-15235--jestjs.netlify.app
📱 Preview on mobile
Toggle QR Code...

QR Code

Use your smartphone camera to open QR code link.

To edit notification comments on pull requests, go to your Netlify site configuration.

@SimenB
Copy link
Member

SimenB commented Aug 3, 2024

Right, this seems reasonable. Can we add a test as well?

@connectdotz
Copy link
Contributor

wouldn't the following block, where it checks for a relative path, suffer the same problem for windows relative paths?

@brandonchinn178
Copy link
Contributor Author

@connectdotz You mean because it would be .\ instead of ./?

@connectdotz
Copy link
Contributor

@brandonchinn178

You mean because it would be .\ instead of ./?

Yes.

I saw your change, and it should work for most cases. However, I did notice at least one use case where it might not work: a relative path pattern with a leading escape, such as "\\.\\\\subdir\\\\file\\.test\\.ts" (regex for .\subdir\file.test.ts). I think even path.isAbsolute() might fail on this one (it would return true instead of false).

It seems the complexity here is due to the fact that we are dealing with a regex pattern for the path, not the path itself.

In any case, our extension doesn't use relative path patterns, so this may not be relevant for us. However, I wanted to bring this potential issue to your attention so you can decide how (or if) you'd like to handle it.

@brandonchinn178
Copy link
Contributor Author

I don't think that's an expected usage. We use a regex to match, yes, but the user input isnt a regex; in isMatch, we escape the entire path to make it a literal regex.

I think I tried doing something originally like "strip root directory from the test filename, then check if filter is a substring in test filename" and not use regex at all, but I think it broke something else...

@connectdotz
Copy link
Contributor

We use a regex to match, yes, but the user input isn't a regex

Doesn't --testPathPatterns take regex input strings?

However, I do agree that to match a file like c:\whatever\subdir\file.test.ts with a partial regex pattern, one should use "\\\\subdir\\\\file\\.test\\.ts", without the leading ".", with or without the escape. In other words, I'm not sure we need to handle the .${path.sep} pattern at all here; the match should fail if people pass in such a regex string.

That said, we should probably model the implementation using the current jest --testPathPattern as guidance to ensure it is as backward compatible as possible.

@brandonchinn178
Copy link
Contributor Author

@connectdotz You're right, it is supposed to be a regex. This is what I get for skimming PRs on my phone 😄

However, your example here:

a relative path pattern with a leading escape, such as "\.\\subdir\\file\.test\.ts" (regex for .\subdir\file.test.ts)

I don't think this ever worked. I just tested this in a new project with jest 29. The reason that ./foo.spec.js worked is IMO a bit of an accident. The old jest behavior just converted it into a regex verbatim (/.\/foo.spec.js/i) and then matched it against the full path (/home/jdoe/my-project/foo.spec.js) and the ./ matched the end of projec(t/).

@SimenB You're right, I just took another look at the docs.

I think the problem here is that this option tries to be both a path and a regex. This has some odd behavior:

$ cd /home/jdoe/my-project/
$ tree .
.
|-- foo.spec.js
|-- dir1/
    |-- dir2/
        |-- bar.spec.js

$ jest './foo.spec.js'
Path: /home/jdoe/my-project/foo.spec.js           => MATCH (t/foo.spec.js)
Path: /home/jdoe/my-project/dir1/dir2/bar.spec.js => NO MATCH

$ jest '\./foo.spec.js'
Path: /home/jdoe/my-project/foo.spec.js           => NO MATCH
Path: /home/jdoe/my-project/dir1/dir2/bar.spec.js => NO MATCH

$ jest '../foo.spec.js'
Path: /home/jdoe/my-project/foo.spec.js           => MATCH (ct/foo.spec.js)
Path: /home/jdoe/my-project/dir1/dir2/bar.spec.js => NO MATCH

$ jest 'dir1/../foo.spec.js'
Path: /home/jdoe/my-project/foo.spec.js           => NO MATCH
Path: /home/jdoe/my-project/dir1/dir2/bar.spec.js => NO MATCH

$ jest './bar.spec.js'
Path: /home/jdoe/my-project/foo.spec.js           => NO MATCH
Path: /home/jdoe/my-project/dir1/dir2/bar.spec.js => MATCH (2/bar.spec.js)

@SimenB Can I propose we have the following semantics:

  1. If the path is absolute, match it as a regex to the full test path
  2. Otherwise, strip the rootDir from the test path and match the pattern to the rest

This most closely matches the spirit of the option, it seems like. The only breaking change here is that ./foo.spec.js will no longer work, but that seems more logically consistent with the fact that a period everywhere else is a regex dot, not a directory dot.

The alternative is to special case the leading ./ on POSIX (and both ./ and .\ on WINDOWS), but I'm not sure what the best "special case" behavior is here:

  1. Should a leading ./ just be stripped? So ./foo.spec.js would match any foo.spec.js
  2. Should it force resolution at rootDir? This is a breaking change where ./bar.spec.js would no longer work

@SimenB
Copy link
Member

SimenB commented Aug 6, 2024

I would have preferred them being globs rather than regexes, but I guess it's way to late to change that now 😅

jest ./some-local.test.js needs to work as that's a very natural way to define a file path. So I guess your alternative, with 2 is the way to go?

2. This is a breaking change where ./bar.spec.js would no longer work

Seems fine - you're passing a relative path, it should probably only match relatively to where the command is used?

@SimenB
Copy link
Member

SimenB commented Aug 7, 2024

Latest changes look promising! @connectdotz could you give them a quick test?

@connectdotz
Copy link
Contributor

connectdotz commented Aug 8, 2024

@connectdotz could you give them a quick test?

Yes, it works with our use case. 👍

Copy link
Member

@SimenB SimenB left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

thanks!

@SimenB SimenB merged commit 9b08aa0 into jestjs:main Aug 8, 2024
73 checks passed
@brandonchinn178 brandonchinn178 deleted the patch-1 branch August 9, 2024 03:06
Copy link

github-actions bot commented Sep 9, 2024

This pull request has been automatically locked since there has not been any recent activity after it was closed. Please open a new issue for related bugs.
Please note this issue tracker is not a help forum. We recommend using StackOverflow or our discord channel for questions.

@github-actions github-actions bot locked as resolved and limited conversation to collaborators Sep 9, 2024
Sign up for free to subscribe to this conversation on GitHub. Already have an account? Sign in.
Labels
None yet
Projects
None yet
Development

Successfully merging this pull request may close these issues.

[Bug]: v30-alpha.5 TestPathPatterns pattern-match regression
3 participants